Applying user-defined volume-delay functions

The file storing a user-defined VDF must include a specific interface in order to be recognized by Visum. Most importantly, each file *.dll must export the functions listed below, so that Visum can call the *.dll file correctly during assignment.

Preparing a *.dll file for a user-defined VDF

Visum comes with a header file UserDefinedVDF.h which can be used directly for implementing the required functions in C++. The header file and a complete sample project for Microsoft Visual Studio can be found in ...programs\PTV Vision\Data\UserDefVDF. Generally any programming language can be used, as long as it can produce a Windows *.dll file which exposes an interface equivalent to the C++ function declarations.

The functions must have exactly the same signature as in UserDefinedVDF.h. You may not change the return value or remove or swap parameters.

Note: The volume-delay function needs to be continuous and monotonously rising with respect to volumes. This applies to all possible values of the parameters.

Mandatory functions in the *.dll

Declaration

char Init ();

Description

This function is called by Visum once immediately after start-up and before the first use of any of the other functions.

Note

Use this function to reset your data structures or perform any other preparatory functions, if necessary.

Parameters

None

Return value

true – reset successful

false – reset failed; the *.dll file should not be called.

Declaration

void Destroy ();

Description

This function is called by Visum once immediately before shut-down and after the last use of any of the other functions.

Note

Use this function to perform any clean-up, if necessary. For example, free up dynamically allocated memory.

Parameters

None

Return value

None

Declaration

char IsThreadSafe();

Description

This function is called by Visum once immediately after start-up and before the first call of any of the other functions. The returned value indicates whether the Calc... functions listed below have been implemented for simultaneous use. If so, the function can be called once more before the previous call has been finished.

Note

This parallelism can be used in case of multithread implementation of assignment procedures in Visum.

Parameters

None

Return value

true – reentrant

false – not re-entrant

Declaration

char DependsOnTSys ();

Description

This function is called by Visum once immediately after start-up and before the first call of any of the other functions. Use this function to indicate whether your function makes use of separate volumes per transport system or whether volumes are required per transport system. If this is not the case, Visum can avoid extracting and passing these volumes to the Calc… functions and the function does not have to be called for every transport system, which saves calculation time.

If the function is used for connectors, the value 1 must be returned (see return value).

For (main) turns and (main) nodes, the value 0 must be returned (see return value).

Parameters

None

Return value

2 - requires volumes per transport system, but returns the same result for all transport systems

1 - requires volumes per transport system and potentially returns a different result for different transport systems

0 - uses only the total volume in PCU and returns the same result for all transport systems

Declaration

const wchar_t * GetName (const char * langid);

Description

Returns a readable name for the functional form which is used as the list entry in the VDF selection list.

Parameters

langid is a language code which can be used to optionally translate the name to other languages.

Currently, the following options are provided.

‘ENG’ – English

‘DEU’ – German

‘FRA’ – French

‘ITA’ – Italian

‘POL’ – Polish

‘ESP’ – Spanish

‘CHI’ – Chinese

‘JAP’ - Japanese

Notes

If not required, you may ignore this functionality. If you use the language code, always include the case of an unknown code, because languages may be added in the future without explicit notice.

Return value

The readable name of the functional form as a 16-bit UTF-8 character string. The value must be returned as UTF-8 to accommodate special characters in some languages, notably the Asiatic ones. Visum expects a pointer to a string stored in the *.dll file.

Note

Be sure not to allocate the string on the stack, so that the address is still valid after the function call.

Declaration

const char * GetID ();

Description

Should return a string to be used as a unique ID for the functional form.

This ID is stored internally in the version file to record the user allocation to connectors and types of links, nodes or turns, and it serves as the ID of the functional form in the *.xml format for procedure parameters.

Parameters

None

Return value

The ID string as an ASCII string. The string must contain only the characters 0..9, a..z, A..Z. Visum expects a pointer to a string stored in the *.dll file.

Note

Be sure not to allocate the string on the stack, so that the address is still valid after the function call.

Declaration

int GetInterfaceVersion();

Description

The interface definition is versioned for files of the *.dll type, so that function declarations can be changed or extended in the future. Return the version number of the header file for which you program your functions. Visum compares the returned number to the version numbers it knows about, calls the dll functions accordingly or returns an error message if the interface version is not supported.

Parameters

None

Return value

the version number

Declaration

void SetTsysInfo (int numtsys, const wchar_t * tsysids[])

Description

Visum calls this function each time the set of private transport systems changes. As parameters, it passes the number of transport systems and an array of the transport system codes. For efficiency reasons, Visum passes the transport system as a number to the Calc functions, i.e. as the 0-based index into the tsysids array.

Note

To avoid string comparisons in these frequently called functions, you should - in SetTsysInfo - evaluate the numerical TSys index of transport systems that require special treatment in volume-delay functions and store this data.

Parameters

numtsys – the number of transport systems (= length of array tsysids)

tsysids – array of 16-bit UTF-8 character strings, each of which is the value of the attribute Code for one transport system used in the assignment.

Return value

No

Declaration

double Calc (int tsysind, bool tsysisopen, int typ, int numlanes, double length, double cap, double v0, double t0, double gradient, double pcuvol, double vehvolsys[], int uval1, int uval2, int uval3, int uvaltsys, double para_a, double para_b, double para_c, double para_d, double para_f, double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)

Description

The implementation of the volume-delay function itself. Visum calls this function in order to calculate the current travel time tCur by link / turn / connector / node and transport system.

Note

Care should be taken to code the function in a computationally efficient form, because it will be called very frequently.

Parameters

For the network objects, various parameters are passed.

Note

For details, please refer to the table below.

Return value

tCur in [s]

Depending on the network object type, Visum passes the following parameters to the Calc function.

 

Link

Connector

Turn

Node

int tsysind

Zero-based index of the transport system (refers to the array tsysids passed in SetTsysInfo) for which tCur should be computed

bool tsysisopen

Is the network object open for the transport system

int typ

0..99

0..9

0..9

0..99

int numlanes

Number of lanes

1 (arbitrary)

1 (arbitrary)

1 (arbitrary)

double length

Length

[short length]

Length

[short length]

0 (arbitrary)

0 (arbitrary)

double cap

CapPrT [pcu]

10E10 or CapPrT[pcu] if connectors have specified shares for the total origin/destination demand.

CapPrT [pcu]

CapPrT [pcu]

double v0

v0 [m/s]

Length/t0 (or 0 if t0=0)

0

0

double t0

Length/v0 [s] or 10E10 if v0=0

t0 [s]

double gradient

Slope

0 (arbitrary)

0 (arbitrary)

0 (arbitrary)

double pcuvol

VolPCU[pcu] as a linear combination of all TSys volumes in [veh], multiplied by the TSys-specific value of the attribute PCU.

Note

Recommended for most of the applications.

double vehvolsys[]

Alternatively, for non-standard PCU calculations, an array of all TSys volumes in [veh]. The order of entries complies with the order of entries in the tsysids array which is passed to SetTsysInfo.

double uval1/2/3

Values of the attributes AddValue1, AddValue2 and AddValue3

double uvaltsys

Value of the TSys-specific attribute AddValue-TSys

0

double para_a..f2

in the Volume-delay function parameters window

double satcrit

in the Volume-delay function parameters window (Selecting a VDF and setting the parameters)

Optional functions in the *.dll

Declaration

double CalcDerivative (int tsysind, bool tsysisopen, int typ, int numlanes, double length, double cap, double v0, double t0, double gradient, double pcuvol, double vehvolsys[], int uval1, int uval2, int uval3, int uvaltsys, double para_a, double para_b, double para_c, double para_d, double para_f, double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)

Description

The program calls this function in order to calculate the derivative of the current travel time tCur with respect to saturation by link / turn / connector / node and TSys. This function is only called within the bicriterial toll assignment methods TRIBUT - Equilibrium assignment and TRIBUT - Equilibrium_Lohse.

Notes

Care should be taken to code the function in a computationally efficient form, because it will be called very frequently.

If the function has not been implemented in the *.dll, Visum will compute the derivative numerically. However, this requires more time than supplying the CalcDerivative() in the *.dll.

Parameters

Same as for Calc()

Return value

Derivative of tCur in [s]

Declaration

double CalcIntegral (int tsysind, bool tsysisopen, int typ, int numlanes, double length, double cap, double v0, double t0, double gradient, double pcuvol, double vehvolsys[], int uval1, int uval2, int uval3, int uvaltsys, double para_a, double para_b, double para_c, double para_d, double para_f, double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)

Description

The program calls this function in order to calculate the integral of the current travel time tCur with respect to saturation by link / turn / connector / node and TSys.

Note

This function is no longer required. Thus it does not have to be implemented.

If the function is not implemented in the *.dll, Visum will compute the integral numerically. However, this will require more time than supplying CalcIntegral() in the *.dll.

Parameters

Same as for Calc()

Return value

Integral from 0 to saturation of the volume-delay function in [s].

Alternative interface with specification of static attributes in the *.dll

The Calc function is called with a number of network object-dependent attributes such as type, number of lanes, etc., but this does not allow access to other attributes such as user-defined attributes. For this case, three additional, optional functions can be implemented in the *.dll, which allow access to any numerical attributes of the network object. These attributes must be static within an assignment, i.e., they must not depend on the volume. Access to volume-dependent attributes leads to incorrect values. If the following three functions are implemented, the alternative interface is used. The functions Calc(), CalcDerivative() and CalcIntegral() should not be implemented in this case.

Declaration

int GetNumStaticAttributes()

Description

This function returns the number of static attributes for which values are passed to CalculateWithStaticAttributes().

Parameters

None

Return value

Number of static attributes

 

Declaration

const wchar_t * GetStaticAttributeID(int attributesIndexZeroBased)

Description

Should return a string that contains a valid attributeID

Parameters

The index of the attribute (0 to GetNumStaticAttributes()-1)

Return value

The attributeID string as an ASCII string. Visum expects a pointer to a string stored in the *.dll file.

Note

Be sure not to allocate the string on the stack, so that the address is still valid after the function call.

 

Declaration

double CalculateWithStaticAttributes(int tsysind, char tsysisopen, double cap, double t0, pcuvol, double basevol, double vehvolsys[], double staticAttributeValues[],

double para_a, double para_b, double para_c, double para_d, double para_f, double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)

Description

The implementation of the volume-delay function itself. Visum calls this function in order to calculate the current travel time tCur by link / turn / connector / node and transport system.

Note

Care should be taken to code the function in a computationally efficient form, because it will be called very frequently.

Parameters

For the network objects, various parameters are passed. The parameters correspond to those of the function Calc(), but type, numlanes, length, v0, uval1, uval2, uval3, uvaltsys are missing.

Instead in staticAttributeValues an array of the length GetNumStaticAttributes() is passed with the values specified by GetStaticAttributeID(). If an attribute does not exist or the attribute is not numeric, 0 is passed.

Return value

tCur in [s]

Example for a user-defined VDF

Assume that two transport systems CAR and HGV are used in the assignment.

  • For CAR the volume-delay function is a linear form with one breakpoint at satcrit.

  • For HGV two breakpoints d and e are used.

The following applies.

The derivations are:

Source code of the *.dll file
#include “UserDefinedVDF.h”
#include “tchar.h”
int indHGV;  // index of the HGV tsys
wchar_t VDFName[] = _T(“ManualExample”); // UTF-8 !!
char  VDFID[] = “MANEX”;
int INTERFACE_VERSION = 1;
boolchar Init()
{
	indHGV = -1;
	return true;
}
void Destroy()
{
}
char DependsOnTSys()
{
  return 1;
}
const wchar_t* GetName(const char *langid)
{
	return VDFName;
}
const char* GetID(const char *langid)
{
	return VDFID;
}
int GetInterfaceVersion()
{
	return INTERFACE_VERSION;
}
void SetTsysInfo (int numtsys, const wchar_t * tsysids[])  
{
	for (int i = 0; i < numtsys; i++)
	{
		if (_tcscmp(tsysids[i], _T(“HGV”)) == 0) indHGV = i;
	}
}
double Calc (int tsysind, bool tsysisopen, int typ, int numlanes, double length, double cap, double v0, double t0, double gradient, double pcuvol, double vehvolsys[], int uval1, int uval2, int uval3, int uvaltsys, double para_a, double para_b, double para_c, double para_d, double para_f, double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)
{ 
	double sat = pcuvol / cap;
	if (tsysind != indHGV) {
		if (sat < satcrit)
			return t0 * (1 + para_a * sat);
		else
			return t0 * (1 + para_a * satcrit + para_b * (sat-satcrit));
}
	else {
		if (sat < d)
			return t0 * (1 + para_a2 * sat);
		else {
		   if (d <= sat && sat < e)
			  return t0 * (1 + para_a2 * d + para_b2 * (sat-d));
		else
		    return t0 * (1 + para_a2 * d + para_b2*(e-d) + para_c*(sat-e));
		}
	}
}
double CalcDerivative (int tsysind, bool tsysisopen, int typ, int numlanes, double length, double cap, double v0, double t0, double gradient, double pcuvol, double vehvolsys[], int uval1, int uval2, int uval3, int uvaltsys, double para_a, double para_b, double para_c, double para_d, double para_f, double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)
{ 
	double sat = pcuvol / cap;
	if (tsysind != indHGV) {
		if (sat < satcrit)
			return para_a;
		else
			return para_b;
	}
	else {
		if (sat < d)
			return para_a2;
		else {
			if (d <= sat && sat < e)
				return para_b2;
			else
				return para_c;
		}
	}
}
Example of a user-defined VD function with static attributes

The implementation of a BPR function to show access to static attributes.

Source code of the *.dll file
#include "UserDefinedVDF.h"
#include "tchar.h"
#include <math.h>
#include <float.h>
 
wchar_t VDFName[] = _T("ExampleStaticAttributes");
char  VDFID[] = "PTV_EXAMPLESTATICATTRIBUTES";
int INTERFACE_VERSION = 1;
 
char Init()
{
return true;
}
 
enum MyStaticIDs
{
V0PrT,
Length,
CapPrT,
TypeNo,
LastId
};
 
static const int MyMaxIDLength = 100;
static wchar_t staticAttributeIDs[LastId][MyMaxIDLength] =
{
L"V0PRT",
L"LENGTH",
L"CAPPRT",
L"TYPENO"
};
 
int GetNumStaticAttributes()
{
return LastId;
}
 
const wchar_t * GetStaticAttributeID(int attributesIndexZeroBased)
{
return staticAttributeIDs[attributesIndexZeroBased];
}
 
void Destroy()
{
}
 
char IsThreadSafe()
{
return true;
}
 
 
char DependsOnTSys()
{
return 0;
}
 
const wchar_t* GetName(const char *langid)
{
return VDFName;
}
 
const char* GetID()
{
return VDFID;
}
 
int GetInterfaceVersion()
{
return INTERFACE_VERSION;
}
 
void SetTsysInfo (int numtsys, const wchar_t * tsysids[])
{
}
 
double CalculateWithStaticAttributes(int tsysind, char tsysisopen, double cap, double t0,
double pcuvol, double basevol, double vehvolsys[],
double staticAttributeValues[],
double para_a, double para_b, double para_c, double para_d, double para_f,
double para_a2, double para_b2, double para_d2, double para_f2, double satcrit)
{
double const v0PrT = staticAttributeValues[V0PrT];
double const length = staticAttributeValues[Length];
double const mycap = staticAttributeValues[CapPrT];
double const typeNo = staticAttributeValues[TypeNo];
 
if (cap <= 0 || para_c <= 0 || v0PrT <= 0 || typeNo < 0) {
return DBL_MAX;
}
double sat = pcuvol / (cap * para_c);
 
double myt0 = 3.6 * length / v0PrT;
return myt0 * (1 + para_a * (pow(sat, para_b)));
}
 
Importing user-defined VDFs

1.  Compile and link the *.dll with the development environment of your choice.

2.  Name the library file according to the following pattern: VisumVDF***.dll.

Notes: Here, *** can be replaced by any string that matches the Windows file name conventions.

Note that the IDs returned by GetID() must be unique. If two *.dll files use the same ID, only the first one will be loaded into Visum. All others will be ignored.

3.  Store the file in the folder ...\User\<User>\AppData\Roaming\PTV Vision\PTV Visum 2025\UserVDF-DLLs.

Notes: During the installation process, this folder is created by default. It is evaluated, whenever a program session starts.

Alternatively, you may select a different folder as standard folder for your *.dll files. In this case, the different path must be saved to the standard path file std.pfd (Editing the storage location of files). The modification takes effect once you restart the program.

If you change the project directory for user-defined VD functions, the contained *.dll files will be loaded additionally and the existing ones will remain.

During the program start, Visum detects the *.dll files and loads the contents. Only those files are opened, whose Windows edition (32 or 64 Bit) complies with the current Visum edition. Then, the user-defined VDFs are listed jointly with the predefined VDFs in the Volume-delay function parameters window.

Notes: After start-up Visum no longer scans for new *.dll files. If you add another user-defined VDF, you have to start Visum again.

If you store a *.bmp file and the *.dll file with identical file names in the same folder, the graph of the selected VDF is displayed in the Volume-delay function parameters window.

Creating user-defined VD functions with VisualStudio C++

In the installation directory of Visum in the subfolder Data\UserDefVDF you will find a sample environment for Visual Studio 2019.

Note: Before use, you should copy the contents of Data\UserDefVDF to a separate folder outside of C:\Program Files, because this folder Visum can be overwritten in case of an update.

Both the debug and release configuration create *.dll files that are compatible with Visum. However, the release variant is faster at runtime.

  • ►  Start Visual Studio 2019 and open the solution file UserDefVDF.sln..

To create your own *.dll , you can either modify one of the existing projects or add a new project to the solution.